1   package org.infinispan.configuration.cache;
2   
3   import static org.infinispan.configuration.cache.TransactionConfiguration.*;
4   
5   import java.lang.invoke.MethodHandles;
6   import java.util.concurrent.TimeUnit;
7   
8   import javax.transaction.Synchronization;
9   import javax.transaction.TransactionManager;
10  import javax.transaction.xa.XAResource;
11  
12  import org.infinispan.commons.configuration.Builder;
13  import org.infinispan.commons.configuration.attributes.Attribute;
14  import org.infinispan.commons.configuration.attributes.AttributeSet;
15  import org.infinispan.configuration.global.GlobalConfiguration;
16  import org.infinispan.transaction.LockingMode;
17  import org.infinispan.transaction.TransactionMode;
18  import org.infinispan.transaction.TransactionProtocol;
19  import org.infinispan.transaction.lookup.TransactionManagerLookup;
20  import org.infinispan.transaction.lookup.TransactionSynchronizationRegistryLookup;
21  import org.infinispan.util.logging.Log;
22  import org.infinispan.util.logging.LogFactory;
23  
24  /**
25   * Defines transactional (JTA) characteristics of the cache.
26   *
27   * @author pmuir
28   * @author Pedro Ruivo
29   */
30  public class TransactionConfigurationBuilder extends AbstractConfigurationChildBuilder implements Builder<TransactionConfiguration> {
31     private static final Log log = LogFactory.getLog(MethodHandles.lookup().lookupClass());
32     private final AttributeSet attributes;
33     private final RecoveryConfigurationBuilder recovery;
34  
35     TransactionConfigurationBuilder(ConfigurationBuilder builder) {
36        super(builder);
37        attributes = TransactionConfiguration.attributeDefinitionSet();
38        this.recovery = new RecoveryConfigurationBuilder(this);
39     }
40  
41     /**
42      * If the cache is transactional (i.e. {@link #transactionMode(org.infinispan.transaction.TransactionMode)} == TransactionMode.TRANSACTIONAL)
43      * and transactionAutoCommit is enabled then for single operation transactions
44      * the user doesn't need to manually start a transaction, but a transactions
45      * is injected by the system. Defaults to true.
46      */
47     public TransactionConfigurationBuilder autoCommit(boolean b) {
48        attributes.attribute(AUTO_COMMIT).set(b);
49        return this;
50     }
51  
52     /**
53      * If there are any ongoing transactions when a cache is stopped, Infinispan waits for ongoing
54      * remote and local transactions to finish. The amount of time to wait for is defined by the
55      * cache stop timeout. It is recommended that this value does not exceed the transaction timeout
56      * because even if a new transaction was started just before the cache was stopped, this could
57      * only last as long as the transaction timeout allows it.
58      * <p/>
59      * This configuration property may be adjusted at runtime
60      */
61     public TransactionConfigurationBuilder cacheStopTimeout(long l) {
62        attributes.attribute(CACHE_STOP_TIMEOUT).set(l);
63        return this;
64     }
65  
66     /**
67      * If there are any ongoing transactions when a cache is stopped, Infinispan waits for ongoing
68      * remote and local transactions to finish. The amount of time to wait for is defined by the
69      * cache stop timeout. It is recommended that this value does not exceed the transaction timeout
70      * because even if a new transaction was started just before the cache was stopped, this could
71      * only last as long as the transaction timeout allows it.
72      * <p/>
73      * This configuration property may be adjusted at runtime
74      */
75     public TransactionConfigurationBuilder cacheStopTimeout(long l, TimeUnit unit) {
76        return cacheStopTimeout(unit.toMillis(l));
77     }
78  
79     /**
80      * Only has effect for DIST mode and when useEagerLocking is set to true. When this is enabled,
81      * then only one node is locked in the cluster, disregarding numOwners config. On the opposite,
82      * if this is false, then on all cache.lock() calls numOwners RPCs are being performed. The node
83      * that gets locked is the main data owner, i.e. the node where data would reside if
84      * numOwners==1. If the node where the lock resides crashes, then the transaction is marked for
85      * rollback - data is in a consistent state, no fault tolerance.
86      *
87      * @deprecated starting with Infinispan 5.1 single node locking is used by default
88      */
89     @Deprecated
90     public TransactionConfigurationBuilder eagerLockingSingleNode(boolean b) {
91        attributes.attribute(EAGER_LOCKING_SINGLE_NODE).set(b);
92        return this;
93     }
94  
95     /**
96      * Configures whether the cache uses optimistic or pessimistic locking. If the cache is not
97      * transactional then the locking mode is ignored.
98      *
99      * @see org.infinispan.configuration.cache.TransactionConfiguration#transactionMode()
100     */
101    public TransactionConfigurationBuilder lockingMode(LockingMode lockingMode) {
102       attributes.attribute(LOCKING_MODE).set(lockingMode);
103       return this;
104    }
105 
106    LockingMode lockingMode() {
107       return attributes.attribute(LOCKING_MODE).get();
108    }
109 
110    /**
111     * If true, the cluster-wide commit phase in two-phase commit (2PC) transactions will be
112     * synchronous, so Infinispan will wait for responses from all nodes to which the commit was
113     * sent. Otherwise, the commit phase will be asynchronous. Keeping it as false improves
114     * performance of 2PC transactions, but it can lead to inconsistencies when a backup owner
115     * only commits the transaction after the primary owner released the lock.
116     */
117    public TransactionConfigurationBuilder syncCommitPhase(boolean b) {
118       attributes.attribute(SYNC_COMMIT_PHASE).set(b);
119       return this;
120    }
121 
122    /**
123     * See {@link #syncCommitPhase(boolean)}
124     *
125     * @return {@code true} if sync commit phase is enabled
126     */
127    boolean syncCommitPhase() {
128       return attributes.attribute(SYNC_COMMIT_PHASE).get();
129    }
130 
131    /**
132     * If true, the cluster-wide rollback phase in two-phase commit (2PC) transactions will be
133     * synchronous, so Infinispan will wait for responses from all nodes to which the rollback was
134     * sent. Otherwise, the rollback phase will be asynchronous.
135     *
136     * Keeping it as false can lead to inconsistencies when a transaction is rolled back because of
137     * a commit timeout, as a backup owner could commit the transaction after the primary released the lock.
138     */
139    public TransactionConfigurationBuilder syncRollbackPhase(boolean b) {
140       attributes.attribute(SYNC_ROLLBACK_PHASE).set(b);
141       return this;
142    }
143 
144    /**
145     * Configure Transaction manager lookup directly using an instance of TransactionManagerLookup.
146     * Calling this method marks the cache as transactional.
147     */
148    public TransactionConfigurationBuilder transactionManagerLookup(TransactionManagerLookup tml) {
149       attributes.attribute(TRANSACTION_MANAGER_LOOKUP).set(tml);
150       if (tml != null) {
151          this.transactionMode(TransactionMode.TRANSACTIONAL);
152       }
153       return this;
154    }
155 
156    /**
157     * Configure Transaction Synchronization Registry lookup directly using an instance of
158     * TransactionManagerLookup. Calling this method marks the cache as transactional.
159     */
160    public TransactionConfigurationBuilder transactionSynchronizationRegistryLookup(TransactionSynchronizationRegistryLookup lookup) {
161       attributes.attribute(TRANSACTION_SYNCHRONIZATION_REGISTRY_LOOKUP).set(lookup);
162       return this;
163    }
164 
165    public TransactionConfigurationBuilder transactionMode(TransactionMode transactionMode) {
166       attributes.attribute(TRANSACTION_MODE).set(transactionMode);
167       return this;
168    }
169 
170    TransactionMode transactionMode() {
171       if (attributes.attribute(TRANSACTION_MODE).isModified()) {
172          return attributes.attribute(TRANSACTION_MODE).get();
173       } else {
174          return null;
175       }
176    }
177 
178    /**
179     * Prevents more than one transaction being written to a key by enforcing cluster-wide locks
180     * on each write operation. Infinispan attempts to obtain locks on specified cache keys across
181     * all nodes in a cluster. All locks are released during the commit or rollback phase.
182     * This configuration might be used when a high contention on keys is occurring, resulting in
183     * inefficiencies and unexpected roll back operations.
184     *
185     * @deprecated Starting with Infinispan 5.1 eager locking is replaced with pessimistic locking and can
186     * be enforced by setting transaction's locking mode to PESSIMISTIC.
187     */
188    @Deprecated
189    public TransactionConfigurationBuilder useEagerLocking(boolean b) {
190       this.attributes.attribute(USE_EAGER_LOCKING).set(b);
191       return this;
192    }
193 
194    /**
195     * Configures whether the cache registers a synchronization with the transaction manager, or registers itself as an
196     * XA resource. It is often unnecessary to register as a full XA resource unless you intend to make use of recovery
197     * as well, and registering a synchronization is significantly more efficient.
198     * @param b if true, {@link Synchronization}s are used rather than {@link XAResource}s when communicating with a {@link TransactionManager}.
199     */
200    public TransactionConfigurationBuilder useSynchronization(boolean b) {
201       attributes.attribute(USE_SYNCHRONIZATION).set(b);
202       return this;
203    }
204 
205    /**
206     * See {@link #useSynchronization(boolean)}
207     *
208     * @return {@code true} if synchronization enlistment is enabled
209     */
210    boolean useSynchronization() {
211       return attributes.attribute(USE_SYNCHRONIZATION).get();
212    }
213 
214    /**
215     * This method allows configuration of the transaction recovery cache. When this method is
216     * called, it automatically enables recovery. So, if you want it to be disabled, make sure you
217     * call {@link org.infinispan.config.FluentConfiguration.RecoveryConfig#disable()}
218     */
219    public RecoveryConfigurationBuilder recovery() {
220       return recovery;
221    }
222 
223    /**
224     * Before Infinispan 5.1 you could access the cache both transactionally and
225     * non-transactionally. Naturally the non-transactional access is faster and
226     * offers less consistency guarantees. From Infinispan 5.1 onwards, mixed
227     * access is no longer supported, so if you wanna speed up transactional
228     * caches and you're ready to trade some consistency guarantees, you can
229     * enable use1PcForAutoCommitTransactions. <p/>
230     *
231     * What this configuration option does is force an induced transaction,
232     * that has been started by Infinispan as a result of enabling autoCommit,
233     * to commit in a single phase. So only 1 RPC instead of 2RPCs as in the
234     * case of a full 2 Phase Commit (2PC).
235     */
236    public TransactionConfigurationBuilder use1PcForAutoCommitTransactions(boolean b) {
237       attributes.attribute(USE_1_PC_FOR_AUTO_COMMIT_TRANSACTIONS).set(b);
238       return this;
239    }
240 
241    /**
242     *The time interval (millis) at which the thread that cleans up transaction completion information kicks in. Defaults to 30000.
243     */
244    public TransactionConfigurationBuilder reaperWakeUpInterval(long interval) {
245       attributes.attribute(REAPER_WAKE_UP_INTERVAL).set(interval);
246       return this;
247    }
248 
249    /**
250     * The duration (millis) in which to keep information about the completion of a transaction. Defaults to 60000.
251     */
252    public TransactionConfigurationBuilder completedTxTimeout(long timeout) {
253       attributes.attribute(COMPLETED_TX_TIMEOUT).set(timeout);
254       return this;
255    }
256 
257    public TransactionConfigurationBuilder transactionProtocol(TransactionProtocol transactionProtocol) {
258       attributes.attribute(TRANSACTION_PROTOCOL).set(transactionProtocol);
259       return this;
260    }
261 
262    @Override
263    public void validate() {
264       Attribute<Long> reaperWakeUpInterval = attributes.attribute(REAPER_WAKE_UP_INTERVAL);
265       Attribute<Long> completedTxTimeout = attributes.attribute(COMPLETED_TX_TIMEOUT);
266       if (reaperWakeUpInterval.get()< 0)
267          throw log.invalidReaperWakeUpInterval(reaperWakeUpInterval.get());
268       if (completedTxTimeout.get() < 0)
269          throw log.invalidCompletedTxTimeout(completedTxTimeout.get());
270       if(attributes.attribute(TRANSACTION_PROTOCOL).get() == TransactionProtocol.TOTAL_ORDER) {
271          //total order only supports transactional caches
272          if(transactionMode() != TransactionMode.TRANSACTIONAL) {
273             throw log.invalidTxModeForTotalOrder(transactionMode());
274          }
275 
276          //total order only supports replicated and distributed mode
277          if(!clustering().cacheMode().isReplicated() && !clustering().cacheMode().isDistributed()) {
278             throw log.invalidCacheModeForTotalOrder(clustering().cacheMode().friendlyCacheModeString());
279          }
280 
281          if (recovery.create().enabled()) {
282             throw log.unavailableTotalOrderWithTxRecovery();
283          }
284 
285          if (lockingMode() != LockingMode.OPTIMISTIC) {
286             throw log.invalidLockingModeForTotalOrder(lockingMode());
287          }
288       }
289       recovery.validate();
290    }
291 
292    @Override
293    public void validate(GlobalConfiguration globalConfig) {
294       recovery.validate(globalConfig);
295    }
296 
297    @Override
298    public TransactionConfiguration create() {
299       if (attributes.attribute(USE_EAGER_LOCKING).get()) {
300          lockingMode(LockingMode.PESSIMISTIC);
301       }
302       if (transactionMode() == null && getBuilder().invocationBatching().isEnabled())
303          transactionMode(TransactionMode.TRANSACTIONAL);
304       else if (transactionMode() == null)
305          transactionMode(TransactionMode.NON_TRANSACTIONAL);
306       return new TransactionConfiguration(attributes.protect(), recovery.create());
307    }
308 
309    @Override
310    public TransactionConfigurationBuilder read(TransactionConfiguration template) {
311       this.attributes.read(template.attributes());
312       this.recovery.read(template.recovery());
313 
314       return this;
315    }
316 
317    @Override
318    public String toString() {
319       return "TransactionConfigurationBuilder [attributes=" + attributes + ", recovery=" + recovery + "]";
320    }
321 
322 }